<!DOCTYPE html>



  


<html class="theme-next muse use-motion" lang="zh-Hans">
<head>
  <meta charset="UTF-8"/>
<meta http-equiv="X-UA-Compatible" content="IE=edge" />
<meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1"/>
<meta name="theme-color" content="#222">









<meta http-equiv="Cache-Control" content="no-transform" />
<meta http-equiv="Cache-Control" content="no-siteapp" />
















  
  
  <link href="/lib/fancybox/source/jquery.fancybox.css?v=2.1.5" rel="stylesheet" type="text/css" />







<link href="/lib/font-awesome/css/font-awesome.min.css?v=4.6.2" rel="stylesheet" type="text/css" />

<link href="/css/main.css?v=5.1.4" rel="stylesheet" type="text/css" />


  <link rel="apple-touch-icon" sizes="180x180" href="/images/apple-touch-icon-next.png?v=5.1.4">


  <link rel="icon" type="image/png" sizes="32x32" href="/images/favicon-32x32-next.png?v=5.1.4">


  <link rel="icon" type="image/png" sizes="16x16" href="/images/favicon-16x16-next.png?v=5.1.4">


  <link rel="mask-icon" href="/images/logo.svg?v=5.1.4" color="#222">





  <meta name="keywords" content="Hexo, NexT" />










<meta name="description" content="Dubbo在众多的微服务框架中脱颖而出，占据RPC服务框架的半壁江山，非常具有普适性，熟练掌握 Dubbo的应用技巧后深刻理解其内部实现原理，让大家能更好的掌控工作，助力职场，特别能让大家在面试中脱颖而出。 那Dubbo内部的设计理念，实现原理是什么呢？ 本文将结合官方提供的3张图，从如下三个方面介绍其内部的核心实现、以及如何指导实践。">
<meta property="og:type" content="article">
<meta property="og:title" content="Dubbo的设计理念原来就藏在这三张图中">
<meta property="og:url" content="https://www.codingw.net/posts/2e7bc25b.html">
<meta property="og:site_name" content="中间件兴趣圈">
<meta property="og:description" content="Dubbo在众多的微服务框架中脱颖而出，占据RPC服务框架的半壁江山，非常具有普适性，熟练掌握 Dubbo的应用技巧后深刻理解其内部实现原理，让大家能更好的掌控工作，助力职场，特别能让大家在面试中脱颖而出。 那Dubbo内部的设计理念，实现原理是什么呢？ 本文将结合官方提供的3张图，从如下三个方面介绍其内部的核心实现、以及如何指导实践。">
<meta property="og:locale">
<meta property="og:image" content="https://img-blog.csdnimg.cn/20210116210800489.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3ByZXN0aWdlZGluZw==,size_16,color_FFFFFF,t_70#pic_center">
<meta property="og:image" content="https://img-blog.csdnimg.cn/20210116210830727.png#pic_center">
<meta property="og:image" content="https://img-blog.csdnimg.cn/20210116210853206.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3ByZXN0aWdlZGluZw==,size_16,color_FFFFFF,t_70#pic_center">
<meta property="og:image" content="https://img-blog.csdnimg.cn/20210116210919335.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3ByZXN0aWdlZGluZw==,size_16,color_FFFFFF,t_70#pic_center">
<meta property="og:image" content="https://img-blog.csdnimg.cn/20210116210947560.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3ByZXN0aWdlZGluZw==,size_16,color_FFFFFF,t_70#pic_center">
<meta property="og:image" content="https://img-blog.csdnimg.cn/20210117195147360.jpg?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3ByZXN0aWdlZGluZw==,size_16,color_FFFFFF,t_70#pic_center">
<meta property="article:published_time" content="2021-01-17T12:53:08.000Z">
<meta property="article:modified_time" content="2021-04-26T12:54:03.042Z">
<meta property="article:author" content="中间件兴趣圈">
<meta property="article:tag" content="中间件">
<meta name="twitter:card" content="summary">
<meta name="twitter:image" content="https://img-blog.csdnimg.cn/20210116210800489.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3ByZXN0aWdlZGluZw==,size_16,color_FFFFFF,t_70#pic_center">



<script type="text/javascript" id="hexo.configurations">
  var NexT = window.NexT || {};
  var CONFIG = {
    root: '',
    scheme: 'Muse',
    version: '5.1.4',
    sidebar: {"position":"left","display":"post","offset":12,"b2t":false,"scrollpercent":false,"onmobile":false},
    fancybox: true,
    tabs: true,
    motion: {"enable":true,"async":false,"transition":{"post_block":"fadeIn","post_header":"slideDownIn","post_body":"slideDownIn","coll_header":"slideLeftIn","sidebar":"slideUpIn"}},
    duoshuo: {
      userId: '0',
      author: '博主'
    },
    algolia: {
      applicationID: '',
      apiKey: '',
      indexName: '',
      hits: {"per_page":10},
      labels: {"input_placeholder":"Search for Posts","hits_empty":"We didn't find any results for the search: ${query}","hits_stats":"${hits} results found in ${time} ms"}
    }
  };
</script>



  <link rel="canonical" href="https://www.codingw.net/posts/2e7bc25b.html"/>





  <title>Dubbo的设计理念原来就藏在这三张图中 | 中间件兴趣圈</title>
  








<meta name="generator" content="Hexo 5.4.0"></head>

<body itemscope itemtype="http://schema.org/WebPage" lang="zh-Hans">

  
  
    
  

  <div class="container sidebar-position-left page-post-detail">
    <div class="headband"></div>

    <header id="header" class="header" itemscope itemtype="http://schema.org/WPHeader">
      <div class="header-inner"><div class="site-brand-wrapper">
  <div class="site-meta ">
    

    <div class="custom-logo-site-title">
      <a href="/"  class="brand" rel="start">
        <span class="logo-line-before"><i></i></span>
        <span class="site-title">中间件兴趣圈</span>
        <span class="logo-line-after"><i></i></span>
      </a>
    </div>
      
        <p class="site-subtitle">微信搜『中间件兴趣圈』，回复『Java』获取200本优质电子书</p>
      
  </div>

  <div class="site-nav-toggle">
    <button>
      <span class="btn-bar"></span>
      <span class="btn-bar"></span>
      <span class="btn-bar"></span>
    </button>
  </div>
</div>

<nav class="site-nav">
  

  
    <ul id="menu" class="menu">
      
        
        <li class="menu-item menu-item-home">
          <a href="/" rel="section">
            
              <i class="menu-item-icon fa fa-fw fa-home"></i> <br />
            
            首页
          </a>
        </li>
      
        
        <li class="menu-item menu-item-categories">
          <a href="/categories/" rel="section">
            
              <i class="menu-item-icon fa fa-fw fa-question-circle"></i> <br />
            
            分类
          </a>
        </li>
      
        
        <li class="menu-item menu-item-archives">
          <a href="/archives/" rel="section">
            
              <i class="menu-item-icon fa fa-fw fa-question-circle"></i> <br />
            
            归档
          </a>
        </li>
      

      
    </ul>
  

  
</nav>



 </div>
    </header>

    <main id="main" class="main">
      <div class="main-inner">
        <div class="content-wrap">
          <div id="content" class="content">
            

  <div id="posts" class="posts-expand">
    

  

  
  
  

  <article class="post post-type-normal" itemscope itemtype="http://schema.org/Article">
  
  
  
  <div class="post-block">
    <link itemprop="mainEntityOfPage" href="https://www.codingw.net/posts/2e7bc25b.html">

    <span hidden itemprop="author" itemscope itemtype="http://schema.org/Person">
      <meta itemprop="name" content="">
      <meta itemprop="description" content="">
      <meta itemprop="image" content="/images/avatar.gif">
    </span>

    <span hidden itemprop="publisher" itemscope itemtype="http://schema.org/Organization">
      <meta itemprop="name" content="中间件兴趣圈">
    </span>

    
      <header class="post-header">

        
        
          <h1 class="post-title" itemprop="name headline">Dubbo的设计理念原来就藏在这三张图中</h1>
        

        <div class="post-meta">
          <span class="post-time">
            
              <span class="post-meta-item-icon">
                <i class="fa fa-calendar-o"></i>
              </span>
              
                <span class="post-meta-item-text">发表于</span>
              
              <time title="创建于" itemprop="dateCreated datePublished" datetime="2021-01-17T20:53:08+08:00">
                2021-01-17
              </time>
            

            

            
          </span>

          
            <span class="post-category" >
            
              <span class="post-meta-divider">|</span>
            
              <span class="post-meta-item-icon">
                <i class="fa fa-folder-o"></i>
              </span>
              
                <span class="post-meta-item-text">分类于</span>
              
              
                <span itemprop="about" itemscope itemtype="http://schema.org/Thing">
                  <a href="/categories/dubbo/" itemprop="url" rel="index">
                    <span itemprop="name">dubbo</span>
                  </a>
                </span>

                
                
              
            </span>
          

          
            
          

          
          
             <span id="/posts/2e7bc25b.html" class="leancloud_visitors" data-flag-title="Dubbo的设计理念原来就藏在这三张图中">
               <span class="post-meta-divider">|</span>
               <span class="post-meta-item-icon">
                 <i class="fa fa-eye"></i>
               </span>
               
                 <span class="post-meta-item-text">阅读次数&#58;</span>
               
                 <span class="leancloud-visitors-count"></span>
             </span>
          

          
            <span class="post-meta-divider">|</span>
            <span class="page-pv"><i class="fa fa-file-o"></i>
            <span class="busuanzi-value" id="busuanzi_value_page_pv" ></span>次
            </span>
          

          

          

        </div>
      </header>
    

    
    
    
    <div class="post-body" itemprop="articleBody">

      
      

      
        <div id="vip-container"><p>Dubbo在众多的微服务框架中脱颖而出，占据RPC服务框架的半壁江山，非常具有普适性，熟练掌握 Dubbo的应用技巧后深刻理解其内部实现原理，让大家能更好的掌控工作，助力职场，特别能让大家在面试中脱颖而出。</p>
<p>那Dubbo内部的设计理念，实现原理是什么呢？</p>
<p>本文将结合官方提供的3张图，从如下三个方面介绍其内部的核心实现、以及如何指导实践。</p>
<span id="more"></span>

<h2 id="1、服务注册与发现机制"><a href="#1、服务注册与发现机制" class="headerlink" title="1、服务注册与发现机制"></a>1、服务注册与发现机制</h2><p>Dubbo的服务注册与发现机制如下图所示：<br><img src="https://img-blog.csdnimg.cn/20210116210800489.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3ByZXN0aWdlZGluZw==,size_16,color_FFFFFF,t_70#pic_center" alt="在这里插入图片描述"><br>在Dubbo中存在4类角色：</p>
<ul>
<li>Registry<br>注册中心。</li>
<li>Consumer<br>服务调用者、消费端。</li>
<li>Provider<br>服务提供者。</li>
<li>Monitor<br>监控中心。</li>
</ul>
<p>具体的交互流程包括如下关键步骤：</p>
<ol>
<li>服务提供者在启动的时候向注册中心进行注册。</li>
<li>消息消费者在启动的时候向注册中心订阅指定服务，注册中心将以某种机制（推或拉）模式告知消费端服务提供者列表。</li>
<li>当服务提供者数量变化（服务提供者扩容、缩容、宕机等因素），注册中心需要以某种方式(推或拉)告知消费端，以便消费端进行正常的负载均衡。</li>
<li>服务提供者、服务消费者向监控中心汇报TPS等调用数据，以便监控中心进行可视化展示等。</li>
</ol>
<p>Dubbo官方提供了多种注册中心，接下来将以使用最为普遍的Zookeeper进一步介绍注册中心的原理。</p>
<p>首先我们来看一下Zookeeper注册中心中的数据存储目录结构，从目录结构来窥探其实现机制。<br><img src="https://img-blog.csdnimg.cn/20210116210830727.png#pic_center" alt="在这里插入图片描述"><br>Dubbo Zookeeper注册中心，其目录组织结构为 /dubbo/{ServiceName}，再每一个服务名称下会有4个目录：</p>
<ul>
<li>providers<br>服务提供者列表。</li>
<li>consumers<br>消费者列表</li>
<li>routers<br>路由规则列表，关于一个服务可以设置多个路由规则。</li>
<li>configurators<br>动态配置条目。在Dubbo中可以在不重启消费者、服务提供者的前提下动态修改服务提供者、服务消费者的配置，例如修改线程的数量，超时时间等参数。</li>
</ul>
<p>基于Zookeeper注册中心的实现细节如下：</p>
<ol>
<li>服务提供者启动时会向注册中心注册，主要是在对应服务的providers目录下增加一条记录(<strong>临时节点</strong>)，同时监听 configurators节点。</li>
<li>服务消费者启动时会向注册中心订阅，主要是在对应服务的consumers目录下增加一条记录(<strong>临时节点</strong>)，同时监听 configurators、routers 目录。</li>
<li>由于当有新的服务提供者上线后 providers 目录会增加一条记录，消费者能立马收到一个服务提供者列表变化的通知，得以将最新的服务提供者列表推送给服务调用方(消费端)；如果一个服务提供者宕机，由于创建的节点是临时节点，Zookeeper会将该节点移除，同样会触发事件，消费端得知最新的服务提供者列表，从而实现路由的动态注册与发现。</li>
<li>当Dubbo新版本上线后，如果需要进行灰度发布，可以通过dubbo-admin等管理平台添加路由规则，最终会写入到指定服务的router节点(持久节点)，服务调用方会监听该节点的变化，从而感知最新的路由规则，将其用于服务提供者的筛选，从而实现灰度发布等功能。</li>
<li>configurators 节点的运作机制与 router 节点一样，就不重复介绍。</li>
</ol>
<p><strong>扩展思考</strong>：</p>
<p>1、如果注册中心全部宕机，对整个服务体系会有什么影响？</p>
<p>如果整个注册中心全部宕机，整个<strong>服务调用</strong>能正常工作，不会影响现有的服务消费者调用，但消费端无法发现新注册的服务提供者。</p>
<p>2、如果注册中心内存溢出或频繁发生 Full Gc，对整个集群又会带来什么影响呢？</p>
<p>如果频繁发生Full GC，并且如果Full GC的时间超过了Zookeeper会话的过期时间，将会造成<strong>非常严重的影响</strong>，会触发所有临时节点被删除，消费端将无法感知服务提供者的存在，影响服务调用，将大面积抛出 No provider 等错误。<strong>正所谓成也临时节点、败也临时节点</strong>。</p>
<p><strong>为了避免Full Gc带来的严重后果，用于Dubbo注册中心的Zookeeper，一定会要独享，并及时做好内存、CPU等的监控与告警。</strong></p>
<h2 id="2、服务调用"><a href="#2、服务调用" class="headerlink" title="2、服务调用"></a>2、服务调用</h2><p>Dubbo的服务调用设计十分优雅，其实现原理如下图所示：<br><img src="https://img-blog.csdnimg.cn/20210116210853206.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3ByZXN0aWdlZGluZw==,size_16,color_FFFFFF,t_70#pic_center" alt="在这里插入图片描述"><br>服务调用，重点阐述客户端发起一个RPC服务调用时的所有实现细节，包括<strong>服务发现</strong>、<strong>故障转移</strong>、<strong>路由转发</strong>、<strong>负载均衡</strong>等方面，是Dubbo实现灰度发布的理论基础。</p>
<h4 id="2-1-服务发现"><a href="#2-1-服务发现" class="headerlink" title="2.1 服务发现"></a>2.1 服务发现</h4><p>客户端在向服务端发起请求时，首先需要知道的是当前有哪些可用的服务提供者，通常有两种服务发现机制：</p>
<ul>
<li><p>静态化配置<br>不妨回想一下，在Dubbo等微服务框架出现之前，一个模块调用另外一个模块通常的做法是使用一个配置文件，将服务提供的列表配置配置在配置文件中，客户端从按照配置文件中的列表进行沦陷。</p>
<p>其<strong>弊端</strong>也非常明显：如果需要调用的服务众多，配置文件会变得臃肿，对扩容缩容的管理、机器宕机等变更不友好，管理非常困难。</p>
</li>
<li><p>动态发现</p>
<p>通常基于注册中心实现服务的注册与动态发现，由于上文已详细介绍，在这里就不累述。</p>
</li>
</ul>
<h4 id="2-2-负载均衡"><a href="#2-2-负载均衡" class="headerlink" title="2.2 负载均衡"></a>2.2 负载均衡</h4><p>客户端通过服务发现机制，能动态发现当前存活的服务提供者列表，接下来要考虑的是如果从服务提供者列表中选择一个服务提供者发起调用，这就是所谓的<strong>负载均衡</strong>，即 LoadBalance。</p>
<p>在Dubbo中默认提供了随机、加权随机、最少活跃连接、一致性Hash等负载均衡算法。</p>
<h4 id="2-3-路由机制"><a href="#2-3-路由机制" class="headerlink" title="2.3 路由机制"></a>2.3 路由机制</h4><p>其实Dubbo中不仅提供了负载均衡机制，还提供了智能路由机制，这是实现<strong>Dubbo灰度发布</strong>的理论基础。</p>
<p>所谓的路由机制，是在服务提供者列表中，再设置一定的规则，进行过滤选择，负载均衡时只从路由过滤规则筛选出来的服务提供者列表中选择，为了更加形象的阐述路由机制的工作原理，给出如下示意图：<br><img src="https://img-blog.csdnimg.cn/20210116210919335.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3ByZXN0aWdlZGluZw==,size_16,color_FFFFFF,t_70#pic_center" alt="在这里插入图片描述"><br>上述设置了一条路由规则，即查询机构ID为102的查询用户请求信息，请发送到新版本，即192168.3.102上，那主要在进行负载均衡之前先执行路由规则，从原始的服务提供者列表者按照路由规则进行过滤，从中挑选出符合要求的提供者列表，然后再进行负载均衡。</p>
<p><strong>路由机制的核心理念</strong>：在进行<strong>负载均衡之前</strong>先对<strong>服务提供者列表</strong>运用<strong>路由规则</strong>，得出一个参与负载均衡的提供者列表。</p>
<h4 id="2-4-故障转移"><a href="#2-4-故障转移" class="headerlink" title="2.4 故障转移"></a>2.4 故障转移</h4><p>远程服务调用通常涉及到网络等因素，客户端向服务提供者发起RPC请求调用时并不一定100%成功，当调用失败后该采用何种策略呢？</p>
<p>Dubbo提供了如下策略：</p>
<ul>
<li><p>failover<br>失败后选择另外一台服务提供者进行重试，重试次数可配置，通常适合<strong>实现幂等服务的场景</strong>。</p>
</li>
<li><p>failfast</p>
<p>快速失败，失败后立即返回错误。</p>
</li>
<li><p>failsafe<br>调用失败后打印错误日志，返回成功，通常<strong>用于记录审计日志等场景</strong>。</p>
</li>
<li><p>failback<br>调用失败后，返回成功，但会在后台定时无限次重试，重启后不再重试。</p>
</li>
<li><p>forking<br>并发调用，收到第一个响应结果后返回给客户端。通常适合<strong>实时性要求比较高的场景</strong>，但浪费服务器资源，通常可以通过forks参数设置并发调用度。</p>
</li>
</ul>
<h2 id="3、线程派发机制"><a href="#3、线程派发机制" class="headerlink" title="3、线程派发机制"></a>3、线程派发机制</h2><p>Dubbo的通信线程模型入下图所示：<br><img src="https://img-blog.csdnimg.cn/20210116210947560.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3ByZXN0aWdlZGluZw==,size_16,color_FFFFFF,t_70#pic_center" alt="在这里插入图片描述"></p>
<h4 id="3-1-网络通信协议"><a href="#3-1-网络通信协议" class="headerlink" title="3.1 网络通信协议"></a>3.1 网络通信协议</h4><p>网络传输通常需要自定义通信协议，通常采用 <strong>Header + Body 的协议设计理念</strong>，并且 Header 长度固定，并且包含一个长度字段，用于记录整个协议包的大小。</p>
<p>网络传输为了提高传输效率，可以采取对传输数据进行压缩，通常是对 body 进行序列化与压缩。</p>
<p>Dubbo支持目前支持 java、compactedjava、nativejava、fastjson、fst、hessian2、kryo等序列化协议。</p>
<h4 id="3-2-线程派发机制"><a href="#3-2-线程派发机制" class="headerlink" title="3.2 线程派发机制"></a>3.2 线程派发机制</h4><p>在Dubbo中默认会创建200个线程用于处理业务方法，所谓的线程派发机制就是IO线程如何决定何种请求转发到哪类线程中执行。</p>
<p>目前Dubbo中<strong>所有的心跳包、网络读写在IO线程中执行</strong>，无法通过配置进行修改。</p>
<p>Dubbo提供了如下几种线程派发机制(Dispatcher)：</p>
<ul>
<li><p>all<br>所有的请求转发到业务线程池中执行（除IO读写、心跳包）</p>
</li>
<li><p>message<br>只有请求事件在线程池中执行，其他在IO线程上执行。</p>
</li>
<li><p>connection<br>请求事件在线程池中执行，<strong>连接、断开连接事件排队执行（含一个线程的线程池）</strong>。</p>
</li>
<li><p>direct</p>
<p>所有请求直接在IO线程中执行。</p>
</li>
</ul>
<blockquote>
<p>温馨提示：有关线程模型，网络通信模式，可以参考笔者如下这篇文章。</p>
</blockquote>
<p>线程派发机制之所有会有多种策略，主要是考虑<strong>线程切换</strong>带来的开销是否能容忍，即线程切换带来的开销小于多线程处理带来的提升。</p>
<p>例如在Dubbo中，对心跳包只需直接返回PONG包（OK），逻辑非常简单，如果将其转换到业务线程池，并不能带来性能提升，反而因为需要线程切换，带来性能损耗，故在IO线程中直接发送响应包是一个非常可取的做法。</p>
<p>在网络编程中需要遵循一条<strong>最佳实践</strong>：<strong>IO线程中不能有阻塞操作，阻塞操作需要转发到业务线程池</strong>。</p>
<hr>
<p>欢迎关注『中间件兴趣圈』，回复【专栏】获取12个JAVA主流中间件的源码剖析专栏，回复PDF可以获取海量学习资源，快速进阶打怪，实现职场的突破。<br><img src="https://img-blog.csdnimg.cn/20210117195147360.jpg?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3ByZXN0aWdlZGluZw==,size_16,color_FFFFFF,t_70#pic_center" alt="在这里插入图片描述"></p>
</div>

			<script src="https://my.openwrite.cn/js/readmore.js" type="text/javascript"></script>
			<script>
			var isMobile = navigator.userAgent.match(/(phone|pad|pod|iPhone|iPod|ios|iPad|Android|Mobile|BlackBerry|IEMobile|MQQBrowser|JUC|Fennec|wOSBrowser|BrowserNG|WebOS|Symbian|Windows Phone)/i);
			if (!isMobile) {
			    var btw = new BTWPlugin();
			    btw.init({
			        "id": "vip-container",
			        "blogId": "18019-1573088808868-542",
			        "name": "中间件兴趣圈",
			        "qrcode": "https://img-blog.csdnimg.cn/20190314214003962.jpg",
			        "keyword": "more"
			    });
			}
			</script>
		
      
    </div>
    
    
    

    

    

    

    <footer class="post-footer">
      

      
      
      

      
        <div class="post-nav">
          <div class="post-nav-next post-nav-item">
            
              <a href="/posts/4ec081f3.html" rel="next" title="面试官:Netty的线程模型可不是Reactor这么简单">
                <i class="fa fa-chevron-left"></i> 面试官:Netty的线程模型可不是Reactor这么简单
              </a>
            
          </div>

          <span class="post-nav-divider"></span>

          <div class="post-nav-prev post-nav-item">
            
              <a href="/posts/76ab8a6b.html" rel="prev" title="Netty进阶：手把手教你如何编写一个NIO客户端">
                Netty进阶：手把手教你如何编写一个NIO客户端 <i class="fa fa-chevron-right"></i>
              </a>
            
          </div>
        </div>
      

      
      
    </footer>
  </div>
  
  
  
  </article>



    <div class="post-spread">
      
    </div>
  </div>


          </div>
          


          

  



        </div>
        
          
  
  <div class="sidebar-toggle">
    <div class="sidebar-toggle-line-wrap">
      <span class="sidebar-toggle-line sidebar-toggle-line-first"></span>
      <span class="sidebar-toggle-line sidebar-toggle-line-middle"></span>
      <span class="sidebar-toggle-line sidebar-toggle-line-last"></span>
    </div>
  </div>

  <aside id="sidebar" class="sidebar">
    
    <div class="sidebar-inner">

      

      
        <ul class="sidebar-nav motion-element">
          <li class="sidebar-nav-toc sidebar-nav-active" data-target="post-toc-wrap">
            文章目录
          </li>
          <li class="sidebar-nav-overview" data-target="site-overview-wrap">
            站点概览
          </li>
        </ul>
      

      <section class="site-overview-wrap sidebar-panel">
        <div class="site-overview">
          <div class="site-author motion-element" itemprop="author" itemscope itemtype="http://schema.org/Person">
            
              <p class="site-author-name" itemprop="name"></p>
              <p class="site-description motion-element" itemprop="description"></p>
          </div>

          <nav class="site-state motion-element">

            
              <div class="site-state-item site-state-posts">
              
                <a href="/archives/">
              
                  <span class="site-state-item-count">139</span>
                  <span class="site-state-item-name">日志</span>
                </a>
              </div>
            

            
              
              
              <div class="site-state-item site-state-categories">
                <a href="/categories/index.html">
                  <span class="site-state-item-count">18</span>
                  <span class="site-state-item-name">分类</span>
                </a>
              </div>
            

            

          </nav>

          

          

          
          

          
          

          

        </div>
      </section>

      
      <!--noindex-->
        <section class="post-toc-wrap motion-element sidebar-panel sidebar-panel-active">
          <div class="post-toc">

            
              
            

            
              <div class="post-toc-content"><ol class="nav"><li class="nav-item nav-level-2"><a class="nav-link" href="#1%E3%80%81%E6%9C%8D%E5%8A%A1%E6%B3%A8%E5%86%8C%E4%B8%8E%E5%8F%91%E7%8E%B0%E6%9C%BA%E5%88%B6"><span class="nav-number">1.</span> <span class="nav-text">1、服务注册与发现机制</span></a></li><li class="nav-item nav-level-2"><a class="nav-link" href="#2%E3%80%81%E6%9C%8D%E5%8A%A1%E8%B0%83%E7%94%A8"><span class="nav-number">2.</span> <span class="nav-text">2、服务调用</span></a><ol class="nav-child"><li class="nav-item nav-level-4"><a class="nav-link" href="#2-1-%E6%9C%8D%E5%8A%A1%E5%8F%91%E7%8E%B0"><span class="nav-number">2.0.1.</span> <span class="nav-text">2.1 服务发现</span></a></li><li class="nav-item nav-level-4"><a class="nav-link" href="#2-2-%E8%B4%9F%E8%BD%BD%E5%9D%87%E8%A1%A1"><span class="nav-number">2.0.2.</span> <span class="nav-text">2.2 负载均衡</span></a></li><li class="nav-item nav-level-4"><a class="nav-link" href="#2-3-%E8%B7%AF%E7%94%B1%E6%9C%BA%E5%88%B6"><span class="nav-number">2.0.3.</span> <span class="nav-text">2.3 路由机制</span></a></li><li class="nav-item nav-level-4"><a class="nav-link" href="#2-4-%E6%95%85%E9%9A%9C%E8%BD%AC%E7%A7%BB"><span class="nav-number">2.0.4.</span> <span class="nav-text">2.4 故障转移</span></a></li></ol></li></ol></li><li class="nav-item nav-level-2"><a class="nav-link" href="#3%E3%80%81%E7%BA%BF%E7%A8%8B%E6%B4%BE%E5%8F%91%E6%9C%BA%E5%88%B6"><span class="nav-number">3.</span> <span class="nav-text">3、线程派发机制</span></a><ol class="nav-child"><li class="nav-item nav-level-4"><a class="nav-link" href="#3-1-%E7%BD%91%E7%BB%9C%E9%80%9A%E4%BF%A1%E5%8D%8F%E8%AE%AE"><span class="nav-number">3.0.1.</span> <span class="nav-text">3.1 网络通信协议</span></a></li><li class="nav-item nav-level-4"><a class="nav-link" href="#3-2-%E7%BA%BF%E7%A8%8B%E6%B4%BE%E5%8F%91%E6%9C%BA%E5%88%B6"><span class="nav-number">3.0.2.</span> <span class="nav-text">3.2 线程派发机制</span></a></li></ol></li></ol></li></ol></div>
            

          </div>
        </section>
      <!--/noindex-->
      

      

    </div>
  </aside>


        
      </div>
    </main>

    <footer id="footer" class="footer">
      <div class="footer-inner">
        <div class="copyright">&copy; <span itemprop="copyrightYear">2021</span>
  <span class="with-love">
    <i class="fa fa-user"></i>
  </span>
  <span class="author" itemprop="copyrightHolder">中间件兴趣圈</span>

  
</div>


  <div class="powered-by">由 <a class="theme-link" target="_blank" href="https://hexo.io">Hexo</a> 强力驱动</div>



  <span class="post-meta-divider">|</span>



  <div class="theme-info">主题 &mdash; <a class="theme-link" target="_blank" href="https://github.com/iissnan/hexo-theme-next">NexT.Muse</a> v5.1.4</div>




        
<div class="busuanzi-count">
  <script async src="https://dn-lbstatics.qbox.me/busuanzi/2.3/busuanzi.pure.mini.js"></script>

  
    <span class="site-uv">
      <i class="fa fa-user"></i>
      <span class="busuanzi-value" id="busuanzi_value_site_uv"></span>
      
    </span>
  

  
    <span class="site-pv">
      <i class="fa fa-eye"></i>
      <span class="busuanzi-value" id="busuanzi_value_site_pv"></span>
      
    </span>
  
</div>








        
      </div>
    </footer>

    
      <div class="back-to-top">
        <i class="fa fa-arrow-up"></i>
        
      </div>
    

    

  </div>

  

<script type="text/javascript">
  if (Object.prototype.toString.call(window.Promise) !== '[object Function]') {
    window.Promise = null;
  }
</script>









  












  
  
    <script type="text/javascript" src="/lib/jquery/index.js?v=2.1.3"></script>
  

  
  
    <script type="text/javascript" src="/lib/fastclick/lib/fastclick.min.js?v=1.0.6"></script>
  

  
  
    <script type="text/javascript" src="/lib/jquery_lazyload/jquery.lazyload.js?v=1.9.7"></script>
  

  
  
    <script type="text/javascript" src="/lib/velocity/velocity.min.js?v=1.2.1"></script>
  

  
  
    <script type="text/javascript" src="/lib/velocity/velocity.ui.min.js?v=1.2.1"></script>
  

  
  
    <script type="text/javascript" src="/lib/fancybox/source/jquery.fancybox.pack.js?v=2.1.5"></script>
  


  


  <script type="text/javascript" src="/js/src/utils.js?v=5.1.4"></script>

  <script type="text/javascript" src="/js/src/motion.js?v=5.1.4"></script>



  
  

  
  <script type="text/javascript" src="/js/src/scrollspy.js?v=5.1.4"></script>
<script type="text/javascript" src="/js/src/post-details.js?v=5.1.4"></script>



  


  <script type="text/javascript" src="/js/src/bootstrap.js?v=5.1.4"></script>



  


  




	





  





  












  





  

  
  <script src="https://cdn1.lncld.net/static/js/av-core-mini-0.6.4.js"></script>
  <script>AV.initialize("NNEhOL0iOcflg8f1U3HUqiCq-gzGzoHsz", "7kSmkbbb3DktmHALlShDsBUF");</script>
  <script>
    function showTime(Counter) {
      var query = new AV.Query(Counter);
      var entries = [];
      var $visitors = $(".leancloud_visitors");

      $visitors.each(function () {
        entries.push( $(this).attr("id").trim() );
      });

      query.containedIn('url', entries);
      query.find()
        .done(function (results) {
          var COUNT_CONTAINER_REF = '.leancloud-visitors-count';

          if (results.length === 0) {
            $visitors.find(COUNT_CONTAINER_REF).text(0);
            return;
          }

          for (var i = 0; i < results.length; i++) {
            var item = results[i];
            var url = item.get('url');
            var time = item.get('time');
            var element = document.getElementById(url);

            $(element).find(COUNT_CONTAINER_REF).text(time);
          }
          for(var i = 0; i < entries.length; i++) {
            var url = entries[i];
            var element = document.getElementById(url);
            var countSpan = $(element).find(COUNT_CONTAINER_REF);
            if( countSpan.text() == '') {
              countSpan.text(0);
            }
          }
        })
        .fail(function (object, error) {
          console.log("Error: " + error.code + " " + error.message);
        });
    }

    function addCount(Counter) {
      var $visitors = $(".leancloud_visitors");
      var url = $visitors.attr('id').trim();
      var title = $visitors.attr('data-flag-title').trim();
      var query = new AV.Query(Counter);

      query.equalTo("url", url);
      query.find({
        success: function(results) {
          if (results.length > 0) {
            var counter = results[0];
            counter.fetchWhenSave(true);
            counter.increment("time");
            counter.save(null, {
              success: function(counter) {
                var $element = $(document.getElementById(url));
                $element.find('.leancloud-visitors-count').text(counter.get('time'));
              },
              error: function(counter, error) {
                console.log('Failed to save Visitor num, with error message: ' + error.message);
              }
            });
          } else {
            var newcounter = new Counter();
            /* Set ACL */
            var acl = new AV.ACL();
            acl.setPublicReadAccess(true);
            acl.setPublicWriteAccess(true);
            newcounter.setACL(acl);
            /* End Set ACL */
            newcounter.set("title", title);
            newcounter.set("url", url);
            newcounter.set("time", 1);
            newcounter.save(null, {
              success: function(newcounter) {
                var $element = $(document.getElementById(url));
                $element.find('.leancloud-visitors-count').text(newcounter.get('time'));
              },
              error: function(newcounter, error) {
                console.log('Failed to create');
              }
            });
          }
        },
        error: function(error) {
          console.log('Error:' + error.code + " " + error.message);
        }
      });
    }

    $(function() {
      var Counter = AV.Object.extend("Counter");
      if ($('.leancloud_visitors').length == 1) {
        addCount(Counter);
      } else if ($('.post-title-link').length > 1) {
        showTime(Counter);
      }
    });
  </script>



  

  

  
  

  

  

  

</body>
</html>
